home
***
CD-ROM
|
disk
|
FTP
|
other
***
search
/
Language/OS - Multiplatform Resource Library
/
LANGUAGE OS.iso
/
lisp
/
kcl
/
akcl
/
kcl.lha
/
lsp
/
iolib.lsp
< prev
next >
Wrap
Lisp/Scheme
|
1987-06-04
|
7KB
|
200 lines
;; (c) Copyright Taiichi Yuasa and Masami Hagiya, 1984. All rights reserved.
;; Copying of this file is authorized to users who have executed the true and
;; proper "License Agreement for Kyoto Common LISP" with SIGLISP.
;;;; iolib.lsp
;;;;
;;;; The IO library.
(in-package 'lisp)
(export '(with-open-stream with-input-from-string with-output-to-string))
(export '(read-from-string))
(export '(write-to-string prin1-to-string princ-to-string))
(export 'with-open-file)
(export '(y-or-n-p yes-or-no-p))
(export 'dribble)
(in-package 'system)
(proclaim '(optimize (safety 2) (space 3)))
(defmacro with-open-stream ((var stream) . body)
(multiple-value-bind (ds b)
(find-declarations body)
`(let ((,var ,stream))
,@ds
(unwind-protect
(progn ,@b)
(close ,var)))))
(defmacro with-input-from-string ((var string &key index start end) . body)
(if index
(multiple-value-bind (ds b)
(find-declarations body)
`(let ((,var (make-string-input-stream ,string ,start ,end)))
,@ds
(unwind-protect
(progn ,@b)
(setf ,index (si:get-string-input-stream-index ,var)))))
`(let ((,var (make-string-input-stream ,string ,start ,end)))
,@body)))
(defmacro with-output-to-string ((var &optional string) . body)
(if string
`(let ((,var (make-string-output-stream-from-string ,string)))
,@body)
`(let ((,var (make-string-output-stream)))
,@body
(get-output-stream-string ,var))))
(defun read-from-string (string
&optional (eof-error-p t) eof-value
&key (start 0) (end (length string))
preserve-whitespace)
(let ((stream (make-string-input-stream string start end)))
(if preserve-whitespace
(values (read-preserving-whitespace stream eof-error-p eof-value)
(si:get-string-input-stream-index stream))
(values (read stream eof-error-p eof-value)
(si:get-string-input-stream-index stream)))))
(defun write-to-string (object &rest rest
&key escape radix base
circle pretty level length
case gensym array
&aux (stream (make-string-output-stream)))
(declare (ignore escape radix base
circle pretty level length
case gensym array))
(apply #'write object :stream stream rest)
(get-output-stream-string stream))
(defun prin1-to-string (object
&aux (stream (make-string-output-stream)))
(prin1 object stream)
(get-output-stream-string stream))
(defun princ-to-string (object
&aux (stream (make-string-output-stream)))
(princ object stream)
(get-output-stream-string stream))
(defmacro with-open-file ((stream . filespec) . body)
(multiple-value-bind (ds b)
(find-declarations body)
`(let ((,stream (open ,@filespec)))
,@ds
(unwind-protect
(progn ,@b)
(close ,stream)))))
(defun y-or-n-p (&optional string &rest args)
(do ((reply))
(nil)
(when string (format *query-io* "~&~? (Y or N) " string args))
(setq reply (read *query-io*))
(cond ((string-equal (symbol-name reply) "Y")
(return-from y-or-n-p t))
((string-equal (symbol-name reply) "N")
(return-from y-or-n-p nil)))))
(defun yes-or-no-p (&optional string &rest args)
(do ((reply))
(nil)
(when string (format *query-io* "~&~? (Yes or No) " string args))
(setq reply (read *query-io*))
(cond ((string-equal (symbol-name reply) "YES")
(return-from yes-or-no-p t))
((string-equal (symbol-name reply) "NO")
(return-from yes-or-no-p nil)))))
(defun sharp-a-reader (stream subchar arg)
(declare (ignore subchar))
(let ((initial-contents (read stream nil nil t)))
(if *read-suppress*
nil
(do ((i 0 (1+ i))
(d nil (cons (length ic) d))
(ic initial-contents (elt ic 0)))
((>= i arg)
(make-array (nreverse d)
:initial-contents initial-contents))))))
(set-dispatch-macro-character #\# #\a 'sharp-a-reader)
(set-dispatch-macro-character #\# #\A 'sharp-a-reader)
(defun sharp-s-reader (stream subchar arg)
(declare (ignore subchar))
(when (and arg (null *read-suppress*))
(error "~S is an extra argument for the #s readmacro." arg))
(let ((l (read stream)))
(unless (get (car l) 'is-a-structure)
(error "~S is not a structure." (car l)))
;; Intern keywords in the keyword package.
(do ((ll (cdr l) (cddr ll)))
((endp ll)
;; Find an appropriate construtor.
(do ((cs (get (car l) 'structure-constructors) (cdr cs)))
((endp cs)
(error "The structure ~S has no structure constructor."
(car l)))
(when (symbolp (car cs))
(return (apply (car cs) (cdr l))))))
(rplaca ll (intern (string (car ll)) 'keyword)))))
(set-dispatch-macro-character #\# #\s 'sharp-s-reader)
(set-dispatch-macro-character #\# #\S 'sharp-s-reader)
(defvar *dribble-stream* nil)
(defvar *dribble-io* nil)
(defvar *dribble-namestring* nil)
(defvar *dribble-saved-terminal-io* nil)
(defun dribble (&optional (pathname "DRIBBLE.LOG" psp) (f :supersede))
(cond ((not psp)
(when (null *dribble-stream*) (error "Not in dribble."))
(if (eq *dribble-io* *terminal-io*)
(setq *terminal-io* *dribble-saved-terminal-io*)
(warn "*TERMINAL-IO* was rebound while DRIBBLE is on.~%~
You may miss some dribble output."))
(close *dribble-stream*)
(setq *dribble-stream* nil)
(format t "~&Finished dribbling to ~A." *dribble-namestring*))
(*dribble-stream*
(error "Already in dribble (to ~A)." *dribble-namestring*))
(t
(let* ((namestring (namestring pathname))
(stream (open pathname :direction :output
:if-exists f
:if-does-not-exist :create)))
(setq *dribble-namestring* namestring
*dribble-stream* stream
*dribble-saved-terminal-io* *terminal-io*
*dribble-io* (make-two-way-stream
(make-echo-stream *terminal-io* stream)
(make-broadcast-stream *terminal-io* stream))
*terminal-io* *dribble-io*)
(multiple-value-bind (sec min hour day month year)
(get-decoded-time)
(format t "~&Starts dribbling to ~A (~d/~d/~d, ~d:~d:~d)."
namestring year month day hour min sec))))))